define(function (require, exports, module) {
  var Hotbox = require('../hotbox')

  // Nice: http://unixpapa.com/js/key.html
  function isIntendToInput (e) {
    if (e.ctrlKey || e.metaKey || e.altKey) return false

    // a-zA-Z
    if (e.keyCode >= 65 && e.keyCode <= 90) return true

    // 0-9 以及其上面的符号
    if (e.keyCode >= 48 && e.keyCode <= 57) return true

    // 小键盘区域 (除回车外)
    if (e.keyCode != 108 && e.keyCode >= 96 && e.keyCode <= 111) return true

    // 小键盘区域 (除回车外)
    // @yinheli from pull request
    if (e.keyCode != 108 && e.keyCode >= 96 && e.keyCode <= 111) return true

    // 输入法
    if (e.keyCode == 229 || e.keyCode === 0) return true

    return false
  }
  /**
   * @Desc: 下方使用receiver.enable()和receiver.disable()通过
   *        修改div contenteditable属性的hack来解决开启热核后依然无法屏蔽浏览器输入的bug;
   *        特别: win下FF对于此种情况必须要先blur在focus才能解决，但是由于这样做会导致用户
   *             输入法状态丢失，因此对FF暂不做处理
   * @Editor: Naixor
   * @Date: 2015.09.14
   */
  function JumpingRuntime () {
    var fsm = this.fsm
    var minder = this.minder
    var receiver = this.receiver
    var container = this.container
    var receiverElement = receiver.element
    var hotbox = this.hotbox

    // normal -> *
    receiver.listen('normal', function (e) {
      // 为了防止处理进入edit模式而丢失处理的首字母,此时receiver必须为enable
      receiver.enable()
      // normal -> hotbox
      if (e.is('Space')) {
        e.preventDefault()
        // safari下Space触发hotbox,然而这时Space已在receiver上留下作案痕迹,因此抹掉
        if (kity.Browser.safari) {
          receiverElement.innerHTML = ''
        }
        return fsm.jump('hotbox', 'space-trigger')
      }

      /**
       * check
       * @editor Naixor
       * @Date 2015-12-2
       */
      switch (e.type) {
        case 'keydown': {
          if (minder.getSelectedNode() && minder._status !== 'readonly') {
            if (isIntendToInput(e)) {
              return fsm.jump('input', 'user-input')
            };
          } else {
            receiverElement.innerHTML = ''
          }
          // normal -> normal shortcut
          fsm.jump('normal', 'shortcut-handle', e)
          break
        }
        case 'keyup': {
          break
        }
        default: {}
      }
    })

    // hotbox -> normal
    receiver.listen('hotbox', function (e) {
      receiver.disable()
      e.preventDefault()
      var handleResult = hotbox.dispatch(e)
      if (hotbox.state() == Hotbox.STATE_IDLE && fsm.state() == 'hotbox') {
        return fsm.jump('normal', 'hotbox-idle')
      }
    })

    // input => normal
    receiver.listen('input', function (e) {
      receiver.enable()
      if (e.type == 'keydown') {
        if (e.is('Enter')) {
          e.preventDefault()
          return fsm.jump('normal', 'input-commit')
        }
        if (e.is('Esc')) {
          e.preventDefault()
          return fsm.jump('normal', 'input-cancel')
        }
        if (e.is('Tab') || e.is('Shift + Tab')) {
          e.preventDefault()
        }
      } else if (e.type == 'keyup' && e.is('Esc')) {
        e.preventDefault()
        return fsm.jump('normal', 'input-cancel')
      }
    })

    /// ///////////////////////////////////////////
    /// 右键呼出热盒
    /// 判断的标准是：按下的位置和结束的位置一致
    /// ///////////////////////////////////////////
    var downX, downY
    var MOUSE_RB = 2 // 右键

    container.addEventListener('mousedown', function (e) {
      if (e.button == MOUSE_RB) {
        e.preventDefault()
      }
      if (fsm.state() == 'hotbox') {
        hotbox.active(Hotbox.STATE_IDLE)
        fsm.jump('normal', 'blur')
      } else if (fsm.state() == 'normal' && e.button == MOUSE_RB) {
        downX = e.clientX
        downY = e.clientY
      }
    }, false)

    container.addEventListener('mousewheel', function (e) {
      if (fsm.state() == 'hotbox') {
        hotbox.active(Hotbox.STATE_IDLE)
        fsm.jump('normal', 'mousemove-blur')
      }
    }, false)

    container.addEventListener('contextmenu', function (e) {
      e.preventDefault()
    })

    container.addEventListener('mouseup', function (e) {
      if (fsm.state() != 'normal') {
        return
      }
      if (e.button != MOUSE_RB || e.clientX != downX || e.clientY != downY) {
        return
      }
      if (!minder.getSelectedNode()) {
        return
      }
      fsm.jump('hotbox', 'content-menu')
    }, false)

    // 阻止热盒事件冒泡，在热盒正确执行前导致热盒关闭
    hotbox.$element.addEventListener('mousedown', function (e) {
      e.stopPropagation()
    })
  }

  return module.exports = JumpingRuntime
})
